home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Personal Computer World 2009 February
/
PCWFEB09.iso
/
Software
/
Resources
/
Audio, Video & Photo
/
Songbird 0.7.0
/
Songbird_0.7.0_windows-i686-msvc8.exe
/
components
/
sbCommandLine.js
< prev
next >
Wrap
Text File
|
2008-08-06
|
15KB
|
449 lines
/**
//
// BEGIN SONGBIRD GPL
//
// This file is part of the Songbird web player.
//
// Copyright(c) 2005-2008 POTI, Inc.
// http://songbirdnest.com
//
// This file may be licensed under the terms of of the
// GNU General Public License Version 2 (the "GPL").
//
// Software distributed under the License is distributed
// on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either
// express or implied. See the GPL for the specific language
// governing rights and limitations.
//
// You should have received a copy of the GPL along with this
// program. If not, go to http://www.gnu.org/licenses/gpl.html
// or write to the Free Software Foundation, Inc.,
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
//
// END SONGBIRD GPL
//
*/
/**
* \file sbCommandLine.js
* \brief Implementation of the interface nsICommandLine
* \todo Implement the -play functionality
*/
const Cc = Components.classes;
const Ci = Components.interfaces;
const Cr = Components.results;
const SONGBIRD_CLH_CONTRACTID = "@songbirdnest.com/commandlinehandler/general-startup;1?type=songbird";
const SONGBIRD_CLH_CID = Components.ID("{128badd1-aa05-4508-87cc-f3cb3e9b5499}");
const SONGBIRD_CLH_CLASSNAME = "Songbird Command Line Handler";
// "m" for ordinary priority see sbICommandLineHandler.idl
const SONGBIRD_CLH_CATEGORY= "m-songbird-clh";
function resolveURIInternal(aCmdLine, aArgument) {
var uri = aCmdLine.resolveURI(aArgument);
if (!(uri instanceof Components.interfaces.nsIFileURL)) {
return uri;
}
return checkUri(uri, aArgument);
}
function checkUri(aURI, aURL) {
try {
if (aURI instanceof Components.interfaces.nsIFileURL)
if (aURI.file.exists())
return aURI;
}
catch (e) {
Components.utils.reportError(e);
}
// We have interpreted the argument as a relative file URI, but the file
// doesn't exist. Try URI fixup heuristics: see bug 290782.
try {
var urifixup = Components.classes["@mozilla.org/docshell/urifixup;1"]
.getService(Ci.nsIURIFixup);
aURI = urifixup.createFixupURI(aURL, 0);
}
catch (e) {
Components.utils.reportError(e);
}
return aURI;
}
function shouldLoadURI(aURI) {
if (!aURI || aURI == "") return false;
if (aURI && !aURI.schemeIs("chrome"))
return true;
dump("*** Preventing external load of chrome URI into browser window (" + aURI.spec + ")\n");
dump(" Use -chrome <uri> instead\n");
return false;
}
/**
* /brief Songbird commandline handler
*/
function sbCommandLineHandler() {
this.itemHandlers = []; // array of handlers
this.itemUriSpecs = []; // array of uri specs
this.flagHandlers = []; // array of arrays (handler, flag)
this.flags = []; // array of arrays (flag, param)
}
sbCommandLineHandler.prototype = {
// there are specific formatting guidelines for help test, see nsICommandLineHandler
helpInfo : " -test [tests] Run tests on the components listed in the\n" +
" optional comma-separated list of tests.\n" +
" If no tests are passed in ALL tests will be run.\n\n" +
" [url|path] Local path/filename to media items to import and play,\n" +
" or URL to load in the browser.\n\n" +
" -register-extensions Registers extensions and then quits.\n\n",
itemHandlers: null,
itemUriSpecs: null,
flagHandlers: null,
flags: null,
handle : function (cmdLine) {
var urilist = [];
var oldlength = this.itemUriSpecs.length;
if (cmdLine.handleFlag("register-extensions", false)) {
throw Components.results.NS_ERROR_ABORT;
}
try {
var ar;
while ((ar = cmdLine.handleFlagWithParam("url", false))) {
urilist.push(resolveURIInternal(cmdLine, ar));
}
}
catch (e) {
Components.utils.reportError(e);
}
var tests = null;
var emptyParam = false;
try {
tests = cmdLine.handleFlagWithParam("test", false);
}
catch (e) {
// cmdLine throws if there is no param for the flag, but we want the
// parameter to be optional, so catch the exception and let ourselves
// know that things are okay. The flag existed without a param.
emptyParam = true;
}
// if there was a parameter or if we had a flag and no param
if (tests != null || emptyParam) {
// we're running tests, make sure we don't open a window
cmdLine.preventDefault = true;
var testHarness = Cc["@songbirdnest.com/Songbird/TestHarness;1"].getService(Ci.sbITestHarness);
var exception;
try {
testHarness.init(tests);
testHarness.run();
}
catch (e) {
exception = e;
}
var platformStr = Cc["@mozilla.org/system-info;1"]
.getService(Ci.nsIPropertyBag2).getProperty("name");
// If we are on Mac, unfortunately the event-queue slows down to slug
// speed when there isn't a window open and we are shutting down. Since
// there isn't a window being used on these unit tests - the hybrid
// Cocoa/Gecko event loop takes a long time (over 20 minutes) to kill the
// 100 or so threads that get spooled up during the test cases.
// Unfortunately, this also happens on Windows :(
// To fix this problem, we will use a nasty little hack. Open up the a
// plain window that closes itself after a couple of seconds. This fires
// the application shutdown procedure just as if we had closed the main
// Songbird window.
//
// Yes - I know this sucks, I hate myself a little more for doing this.
if (platformStr == "Darwin" ||
platformStr == "Windows_NT") {
var ww = Cc["@mozilla.org/embedcomp/window-watcher;1"]
.getService(Ci.nsIWindowWatcher);
ww.openWindow(null,
"chrome://songbird/content/xul/unitTestShutdownWin.xul",
"shutdownwin", "chrome", null);
}
else {
// Linux can follow this 'standard' shutdown procedure:
//
// Fake the sequence of observer notifications for app shutdown. This
// sequence should match that of canQuitApplication (from
// globalOverlay.js) and nsAppStartup::Quit (from nsAppStartup.cpp).
var os = Cc["@mozilla.org/observer-service;1"].
getService(Ci.nsIObserverService);
// We don't care if anyone tries to cancel quit...
var dummyCancelQuit = Cc["@mozilla.org/supports-PRBool;1"].
createInstance(Ci.nsISupportsPRBool);
os.notifyObservers(dummyCancelQuit, "quit-application-requested", null);
os.notifyObservers(null, "quit-application-granted", null);
var appStartup = Cc["@mozilla.org/toolkit/app-startup;1"]
.getService(Ci.nsIAppStartup);
appStartup.quit(Ci.nsIAppStartup.eAttemptQuit);
}
if (exception) {
throw Cr.NS_ERROR_ABORT;
}
}
// XXX bug 2186
var count = cmdLine.length;
for (var i = 0; i < count; ++i) {
// getArgument sometimes causes an exception for the last parameter,
// even tho i is always below cmdLine.length ! This doesn't seem to
// ever happen when the commandline is starting the app, but seems to
// always do when the app is already started, and the commandline is
// received from another instance.
try {
var curarg = cmdLine.getArgument(i);
} catch (e) {
Components.utils.reportError(e);
// try next argument. there shouldn't be any, but just in case...
continue
}
if (curarg == "") continue;
if (curarg.match(/^-/)) {
// Components.utils.reportError("Warning: unrecognized command line flag " + curarg + "\n");
// To emulate the pre-nsICommandLine behavior, we ignore
// the argument after an unrecognized flag.
++i;
} else {
try {
cmdLine.removeArguments(i, i);
urilist.push(resolveURIInternal(cmdLine, curarg));
}
catch (e) {
Components.utils.reportError("Error opening URI '" + curarg + "' from the command line: " + e + "\n");
}
}
}
for (var uri in urilist) {
if (shouldLoadURI(urilist[uri])) {
this.itemUriSpecs.push(urilist[uri].spec);
}
}
if (this.itemUriSpecs.length > oldlength)
this.dispatchItems();
this.handleRemainingFlags(cmdLine);
this.dispatchFlags();
},
handleURL: function(aURL) {
var ioService = Components.classes["@mozilla.org/network/io-service;1"]
.getService(Components.interfaces.nsIIOService);
var uri = ioService.newURI(aURL, null, null);
uri = checkUri(uri, aURL);
if (shouldLoadURI(uri)) {
this.itemUriSpecs.push(uri.spec);
this.dispatchItems();
}
},
handleRemainingFlags: function(cmdLine) {
while (cmdLine.length) {
var curarg = cmdLine.getArgument(0);
if (curarg.match(/^-/)) {
var flag = curarg.slice(1);
var param;
var emptyParam = false;
try {
param = cmdLine.handleFlagWithParam(flag, false);
}
catch (e) {
// cmdLine throws if there is no param for the flag, but we want the
// parameter to be optional, so catch the exception and let ourselves
// know that things are okay. The flag existed without a param.
emptyParam = true;
cmdLine.handleFlag(flag, false);
}
// if there was a parameter or if we had a flag and no param
if (param != null || emptyParam) {
// record the flag for handling by flag handlers
this.flags.push([flag, param]);
}
} else {
// this should really not occur, because the case should have
// been handled as a play item earlier. however, if for some reason
// it does occur, not doing the following would cause an infinite loop,
// so do it just in case.
cmdLine.removeArguments(0, 0);
}
}
},
addItemHandler: function (aHandler) {
this.itemHandlers.push(aHandler);
// dispatch unhandled items immediatly to this handler
this.dispatchItemsToHandler(aHandler);
},
removeItemHandler: function(aHandler) {
var index = this.itemHandlers.indexOf(aHandler);
if (index != -1) this.itemHandlers.splice(index, 1);
},
dispatchItemsToHandler: function(aHandler) {
var count = 0;
var total = this.itemUriSpecs.length;
for (var i=0; i < this.itemUriSpecs.length; i++) {
if (aHandler.handleItem(this.itemUriSpecs[i], count++, total)) {
this.itemUriSpecs.splice(i--, 1);
}
}
},
dispatchItems: function() {
// The last handler to get registered gets
// priority over the first ones, so that if
// there are several instances of the main window,
// the items open in the one created last.
for (var handleridx = this.itemHandlers.length-1; handleridx >= 0; handleridx--) {
this.dispatchItemsToHandler(this.itemHandlers[handleridx]);
if (this.itemUriSpecs.length == 0) break;
}
},
addFlagHandler: function (aHandler, aFlag) {
var entry = [aHandler, aFlag];
this.flagHandlers.push(entry);
// dispatch unhandled flags immediatly to this handler
this.dispatchFlagsToHandler(entry);
},
removeFlagHandler: function(aHandler, aFlag) {
for (var i=this.flagHandlers.length-1;i>=0;i--) {
var entry = this.flagHandlers[i];
if (entry[0] == aHandler && entry[1] == aFlag) {
this.flagHandlers.splice(i, 1);
return;
}
}
},
dispatchFlagsToHandler: function(aHandlerEntry) {
var handler = aHandlerEntry[0];
var flag = aHandlerEntry[1];
for (var i=0; i < this.flags.length; i++) {
if (this.flags[i][0] == flag) {
if (handler.handleFlag(this.flags[i][0], this.flags[i][1])) {
this.flags.splice(i--, 1);
}
}
}
},
dispatchFlags: function() {
// The last handler to get registered gets
// priority over the first ones, so that if
// there are several instances of the main window,
// the flags are handled by the one created last.
for (var handleridx = this.flagHandlers.length-1; handleridx >= 0; handleridx--) {
this.dispatchFlagsToHandler(this.flagHandlers[handleridx]);
if (this.flags.length == 0) break;
}
},
QueryInterface : function clh_QI(iid) {
if (iid.equals(Ci.nsICommandLineHandler) ||
iid.equals(Ci.sbICommandLineManager) ||
iid.equals(Ci.nsISupports))
return this;
throw Cr.NS_ERROR_NO_INTERFACE;
}
}; // sbCommandeLineHandler
/**
* /brief The module for getting the commandline handler
*/
const sbCommandLineHandlerModule = {
registerSelf : function (compMgr, fileSpec, location, type) {
compMgr.QueryInterface(Ci.nsIComponentRegistrar);
compMgr.registerFactoryLocation(SONGBIRD_CLH_CID,
SONGBIRD_CLH_CLASSNAME,
SONGBIRD_CLH_CONTRACTID,
fileSpec,
location,
type);
var catMan = Cc["@mozilla.org/categorymanager;1"].getService(Ci.nsICategoryManager);
catMan.addCategoryEntry("command-line-handler",
SONGBIRD_CLH_CATEGORY,
SONGBIRD_CLH_CONTRACTID,
true,
true);
},
getClassObject : function (compMgr, cid, iid) {
if (!cid.equals(SONGBIRD_CLH_CID))
throw Cr.NS_ERROR_NO_INTERFACE;
if (!iid.equals(Ci.nsIFactory))
throw Cr.NS_ERROR_NOT_IMPLEMENTED;
return this.mFactory;
},
mFactory : {
createInstance : function (outer, iid) {
if (outer != null)
throw Cr.NS_ERROR_NO_AGGREGATION;
return (new sbCommandLineHandler()).QueryInterface(iid);
}
},
unregisterSelf : function (compMgr, location, type) {
compMgr.QueryInterface(Ci.nsIComponentRegistrar);
compMgr.unregisterFactoryLocation(SONGBIRD_CLH_CID, location);
var catMan = Cc["@mozilla.org/categorymanager;1"].getService(Ci.nsICategoryManager);
catMan.deleteCategoryEntry("command-line-handler", SONGBIRD_CLH_CATEGORY);
},
canUnload : function (compMgr) {
return true;
},
QueryInterface : function (iid) {
if ( !iid.equals(Ci.nsIModule) ||
!iid.equals(Ci.nsISupports) )
throw Cr.NS_ERROR_NO_INTERFACE;
return this;
}
}; // sbCommandLineHandlerModule
function NSGetModule(comMgr, fileSpec)
{
return sbCommandLineHandlerModule;
}